package 算法基础模板.chapter_01.双指针.sliding_window;

/**
 * 904. 水果成篮
 * 在一排树中，第 i 棵树产生 tree[i] 型的水果。
 * 你可以从你选择的任何树开始，然后重复执行以下步骤：
 *
 * 把这棵树上的水果放进你的篮子里。如果你做不到，就停下来。
 * 移动到当前树右侧的下一棵树。如果右边没有树，就停下来。
 * 请注意，在选择一颗树后，你没有任何选择：你必须执行步骤 1，然后执行步骤 2，然后返回步骤 1，然后执行步骤 2，依此类推，直至停止。
 *
 * 你有两个篮子，每个篮子可以携带任何数量的水果，但你希望每个篮子只携带一种类型的水果。
 *
 * 用这个程序你能收集的水果树的最大总量是多少？
 *
 *
 *
 * 示例 1：
 *
 * 输入：[1,2,1]
 * 输出：3
 * 解释：我们可以收集 [1,2,1]。
 * 示例 2：
 *
 * 输入：[0,1,2,2]
 * 输出：3
 * 解释：我们可以收集 [1,2,2]
 * 如果我们从第一棵树开始，我们将只能收集到 [0, 1]。
 * 示例 3：
 *
 * 输入：[1,2,3,2,2]
 * 输出：4
 * 解释：我们可以收集 [2,3,2,2]
 * 如果我们从第一棵树开始，我们将只能收集到 [1, 2]。
 * 示例 4：
 *
 * 输入：[3,3,3,1,2,1,1,2,3,3,4]
 * 输出：5
 * 解释：我们可以收集 [1,2,1,1,2]
 * 如果我们从第一棵树或第八棵树开始，我们将只能收集到 4 棵水果树。
 *
 *
 * 提示：
 *
 * 1 <= tree.length <= 40000
 * 0 <= tree[i] < tree.length
 * @author Summerday
 */
public class LC904 {

    public int totalFruit(int[] tree) {
        int l = 0, r = 0, ans = 0, p = 0, n = tree.length;
        int[] cnt = new int[40010];
        while(r < n){
            if(cnt[tree[r]] == 0){
                p ++;
            }
            cnt[tree[r]] ++;
            // 出现的元素种类 超过2 的时候 考虑缩短左边界
            while(p > 2){
                cnt[tree[l]] --;
                // 一种种类为0了
                if(cnt[tree[l]] == 0){
                    p --;
                }
                l ++;
            }
            ans = Math.max(ans, r - l + 1);
            r ++;
        }
        return ans;
    }
}
